package com.haiyou.common.util.idCard;

import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.Hashtable;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.IntStream;

import org.springframework.util.StringUtils;

import com.haiyou.common.util.date.DateUtil;


public class IdCardUtil {
	/***********************************
	 * 身份证验证开始
	 ****************************************/
	/**
	 * 身份证号码验证 1、号码的结构 公民身份号码是特征组合码，由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为：六位数字地址码，
	 * 八位数字出生日期码，三位数字顺序码和一位数字校验码。 2、地址码(前六位数）
	 * 表示编码对象常住户口所在县(市、旗、区)的行政区划代码，按GB/T2260的规定执行。 3、出生日期码（第七位至十四位）
	 * 表示编码对象出生的年、月、日，按GB/T7408的规定执行，年、月、日代码之间不用分隔符。 4、顺序码（第十五位至十七位）
	 * 表示在同一地址码所标识的区域范围内，对同年、同月、同日出生的人编定的顺序号， 顺序码的奇数分配给男性，偶数分配给女性。 5、校验码（第十八位数）
	 * （1）十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, ... , 16 ，先对前17位数字的权求和
	 * Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
	 * （2）计算模 Y = mod(S, 11) （3）通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9 8
	 * 7 6 5 4 3 2
	 */

	/**
	 * 功能：身份证的有效验证
	 *
	 * @param IDStr 身份证号
	 * @return 有效：返回"" 无效：返回String信息
	 */
	@SuppressWarnings("rawtypes")
	public static String IDCardValidate(String IDStr) {
		try {
			String errorInfo = "";// 记录错误信息
			String[] ValCodeArr = { "1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2" };
			String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2" };
			String Ai = "";
			// ================ 号码的长度 15位或18位 ================
			if (IDStr.length() != 15 && IDStr.length() != 18) {
				errorInfo = "身份证号码长度应该为15位或18位。";
				return errorInfo;
			}
			// =======================(end)========================

			// ================ 数字 除最后以为都为数字 ================
			if (IDStr.length() == 18) {
				Ai = IDStr.substring(0, 17);
			} else if (IDStr.length() == 15) {
				Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
			}
			if (isNumeric(Ai) == false) {
				errorInfo = "身份证15位号码都应为数字 ; 18位号码除最后一位外，都应为数字。";
				return errorInfo;
			}
			// =======================(end)========================

			// ================ 出生年月是否有效 ================
			String strYear = Ai.substring(6, 10);// 年份
			String strMonth = Ai.substring(10, 12);// 月份
			String strDay = Ai.substring(12, 14);// 月份
			if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {
				errorInfo = "身份证生日无效。";
				return errorInfo;
			}
			GregorianCalendar gc = new GregorianCalendar();
			SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
			try {
				if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
						|| (gc.getTime().getTime() - s.parse(strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
					errorInfo = "身份证生日不在有效范围。";
					return errorInfo;
				}
			} catch (NumberFormatException e) {
				e.printStackTrace();
			} catch (java.text.ParseException e) {
				e.printStackTrace();
			}
			if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
				errorInfo = "身份证月份无效";
				return errorInfo;
			}
			if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
				errorInfo = "身份证日期无效";
				return errorInfo;
			}
			// =====================(end)=====================

			// ================ 地区码时候有效 ================
			Hashtable h = GetAreaCode();
			if (h.get(Ai.substring(0, 2)) == null) {
				errorInfo = "身份证地区编码错误。";
				return errorInfo;
			}
			// ==============================================

			// ================ 判断最后一位的值 ================
			int TotalmulAiWi = 0;
			for (int i = 0; i < 17; i++) {
				TotalmulAiWi = TotalmulAiWi + Integer.parseInt(String.valueOf(Ai.charAt(i))) * Integer.parseInt(Wi[i]);
			}
			int modValue = TotalmulAiWi % 11;
			String strVerifyCode = ValCodeArr[modValue];
			Ai = Ai + strVerifyCode;

			if (IDStr.contains("x"))
				IDStr.replaceAll("x", "X");
			if (IDStr.length() == 18) {
				if (Ai.equals(IDStr) == false) {
					errorInfo = "身份证无效，不是合法的身份证号码";
					return errorInfo;
				}
			} else {
				return "";
			}
			// =====================(end)=====================
		} catch (Exception e) {
			return "";
		}

		return "";
	}

	/**
	 * 功能：设置地区编码
	 *
	 * @return Hashtable 对象
	 */
	@SuppressWarnings({ "rawtypes", "unchecked" })
	private static Hashtable GetAreaCode() {
		Hashtable hashtable = new Hashtable();
		hashtable.put("11", "北京");
		hashtable.put("12", "天津");
		hashtable.put("13", "河北");
		hashtable.put("14", "山西");
		hashtable.put("15", "内蒙古");
		hashtable.put("21", "辽宁");
		hashtable.put("22", "吉林");
		hashtable.put("23", "黑龙江");
		hashtable.put("31", "上海");
		hashtable.put("32", "江苏");
		hashtable.put("33", "浙江");
		hashtable.put("34", "安徽");
		hashtable.put("35", "福建");
		hashtable.put("36", "江西");
		hashtable.put("37", "山东");
		hashtable.put("41", "河南");
		hashtable.put("42", "湖北");
		hashtable.put("43", "湖南");
		hashtable.put("44", "广东");
		hashtable.put("45", "广西");
		hashtable.put("46", "海南");
		hashtable.put("50", "重庆");
		hashtable.put("51", "四川");
		hashtable.put("52", "贵州");
		hashtable.put("53", "云南");
		hashtable.put("54", "西藏");
		hashtable.put("61", "陕西");
		hashtable.put("62", "甘肃");
		hashtable.put("63", "青海");
		hashtable.put("64", "宁夏");
		hashtable.put("65", "新疆");
		hashtable.put("71", "台湾");
		hashtable.put("81", "香港");
		hashtable.put("82", "澳门");
		hashtable.put("91", "国外");
		return hashtable;
	}

	/**
	 * 功能：判断字符串是否为数字
	 *
	 * @param str
	 * @return
	 */
	private static boolean isNumeric(String str) {
		Pattern pattern = Pattern.compile("[0-9]*");
		Matcher isNum = pattern.matcher(str);
		if (isNum.matches()) {
			return true;
		} else {
			return false;
		}
	}

	/**
	 * 功能：判断字符串是否为日期格式
	 *
	 * @return
	 */
	public static boolean isDate(String strDate) {
		Pattern pattern = Pattern.compile(
				"^((\\d{2}(([02468][048])|([13579][26]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])))))|(\\d{2}(([02468][1235679])|([13579][01345789]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|(1[0-9])|(2[0-8]))))))(\\s(((0?[0-9])|([1-2][0-3]))\\:([0-5]?[0-9])((\\s)|(\\:([0-5]?[0-9])))))?$");
		Matcher m = pattern.matcher(strDate);
		if (m.matches()) {
			return true;
		} else {
			return false;
		}
	}

	public static boolean isMale(String id) {
		String result = IDCardValidate(id);
		if (StringUtils.isEmpty(result)) {
			if (id.length() == 15) {
				return Integer.parseInt(id.substring(14, 15)) % 2 == 0 ? false : true;
			} else if (id.length() == 18) {
				return Integer.parseInt(id.substring(16, 17)) % 2 == 0 ? false : true;
			}
		}
		return true;
	}

	/***********************************
	 * 身份证验证结束
	 ****************************************/

	/**
	 * 检查 email输入是否正确 正确的书写格 式为 username@domain
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkEmail(String value, int length) {
		return value.matches("\\w+([-+.]\\w+)*@\\w+([-.]\\w+)*\\.\\w+([-.]\\w+)*") && value.length() <= length;
	}

	public static boolean checkMobile(String str) {
		Pattern p = null;
		Matcher m = null;
		boolean b = false;
		p = Pattern.compile("^[1][3,4,5,7,8][0-9]{9}$"); // 验证手机号
		m = p.matcher(str);
		b = m.matches();
		return b;
	}

	/**
	 * 电话号码验证
	 *
	 * @param str
	 * @return 验证通过返回true
	 */
	public static boolean checkTel(String str) {
		Pattern p1 = null, p2 = null;
		Matcher m = null;
		boolean b = false;
		p1 = Pattern.compile("^[0][1-9]{2,3}-?[0-9]{5,10}$"); // 验证带区号的
		p2 = Pattern.compile("^[1-9]{1}[0-9]{5,8}$"); // 验证没有区号的
		if (str.length() > 9) {
			m = p1.matcher(str);
			b = m.matches();
		} else {
			m = p2.matcher(str);
			b = m.matches();
		}
		return b;
	}

	/**
	 * 验证年龄0-120
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkAge(String value) {
		return value.matches("120|((1[0-1]|\\d)?\\d)");
	}

	/**
	 * 检查中文名输 入是否正确
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkChineseName(String value, int length) {
		return value.matches("^[\\u4e00-\\u9fa5.·\\u36c3\\u4DAE]{2,}$") && value.length() <= length;
	}
	
	public static void main(String[] args) {
		
		//String content = "阿布几米吉提·阿布几米提";
		String content = "符文震";
		System.err.println(checkChineseName(content, 50));
		
	}
	

	/**
	 * 检查字符串是 否含有HTML标签
	 *
	 * @param value
	 * @return
	 */

	public static boolean checkHtmlTag(String value) {
		return value.matches("<(\\S*?)[^>]*>.*?</\\1>|<.*? />");
	}

	/**
	 * 检查URL是 否合法
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkURL(String value) {
		return value.matches("[a-zA-z]+://[^\\s]*");
	}

	/**
	 * 检查IP是否 合法
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkIP(String value) {
		return value.matches("\\d{1,3}+\\.\\d{1,3}+\\.\\d{1,3}+\\.\\d{1,3}");
	}

	/**
	 * 检查ID是否 合法，开头必须是大小写字母，其他位可以有大小写字符、数字、下划线
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkID(String value) {
		return value.matches("[a-zA-Z][a-zA-Z0-9_]{4,15}{1");
	}

	/**
	 * 检查QQ是否 合法，必须是数字，且首位不能为0，最长15位
	 *
	 * @param value
	 * @return
	 */

	public static boolean checkQQ(String value) {
		return value.matches("[1-9][0-9]{4,13}");
	}

	/**
	 * 检查邮编是否 合法
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkPostCode(String value) {
		return value.matches("[1-9]\\d{5}(?!\\d)");
	}

	/**
	 * 检查身份证是 否合法,15位或18位
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkIDCard(String value) {
		return value.matches("\\d{15}|\\d{18}");
	}

	/**
	 * 检查输入是否 超出规定长度 Java教程:http://www.javaweb.cc
	 *
	 * @param length
	 * @param value
	 * @return
	 */
	public static boolean checkLength(String value, int length) {
		return ((value == null || "".equals(value.trim())) ? 0 : value.length()) <= length;
	}

	/**
	 * 检查是否为空 字符串,空：true,不空:false
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkNull(String value) {
		return value == null || "".equals(value.trim());
	}

	/**
	 * 检查车牌号 否合法
	 *
	 * @param value
	 * @return
	 */
	public static boolean checkCarPlate(String value) {
		return value.matches("^[\u4e00-\u9fa5]{1}[A-Z]{1}[A-Z_0-9]{4}[\u4e00-\u9fa5_A-Z_0-9]{1}$");
	}

	/**
	 * 获取生日
	 * 
	 * @param idCard
	 * @return
	 */
	public static Date getBirthdays(String idCard) {

		// 出生年
		Integer personYear = Integer.parseInt(idCard.substring(6, 10));
		// 出生月
		Integer personMonth = Integer.parseInt(idCard.substring(10, 12));
		// 出生日
		Integer personDay = Integer.parseInt(idCard.substring(12, 14));

		StringBuffer bp = new StringBuffer();

		bp.append(personYear).append("-");
		bp.append(personMonth).append("-");
		bp.append(personDay);

		return DateUtil.parse(bp.toString(), DateUtil.DATE_FORMAT);

	}

	/**
	 * 根据身份证号计算年龄
	 * 
	 * @param pensonnelIdCard
	 * @return
	 */
	public static Integer getAgeFromIdCard(String pensonnelIdCard) {

		// 截取身份证中出行人出生日期中的年、月、日
		Integer personYear = Integer.parseInt(pensonnelIdCard.substring(6, 10));
		Integer personMonth = Integer.parseInt(pensonnelIdCard.substring(10, 12));
		Integer personDay = Integer.parseInt(pensonnelIdCard.substring(12, 14));

		Calendar cal = Calendar.getInstance();
		// 得到当前时间的年、月、日
		Integer yearNow = cal.get(Calendar.YEAR);
		Integer monthNow = cal.get(Calendar.MONTH) + 1;
		Integer dayNow = cal.get(Calendar.DATE);

		// 用当前年月日减去生日年月日
		Integer yearMinus = yearNow - personYear;
		Integer monthMinus = monthNow - personMonth;
		Integer dayMinus = dayNow - personDay;

		Integer age = yearMinus; // 先大致赋值

		if (yearMinus == 0) { // 出生年份为当前年份
			age = 0;
		} else { // 出生年份大于当前年份
			if (monthMinus < 0) {// 出生月份小于当前月份时，还没满周岁
				age = age - 1;
			}
			if (monthMinus == 0) {// 当前月份为出生月份时，判断日期
				if (dayMinus < 0) {// 出生日期小于当前月份时，没满周岁
					age = age - 1;
				}
			}
		}
		return age;
	}



	/**
	 * 身份证号校验 （支持18位）
	 * 
	 */
	public static boolean checkIdentityCode(String identityCode) {
		if (!identityCode.matches("\\d{17}(\\d|x|X)$")) {
			return false;
		}
		Date d = new Date();
		DateFormat df = new SimpleDateFormat("yyyyMMdd");
		int year = Integer.parseInt(df.format(d));
		if (Integer.parseInt(identityCode.substring(6, 10)) < 1900
				|| Integer.parseInt(identityCode.substring(6, 10)) > year) {// 7-10位是出生年份，范围应该在1900-当前年份之间
			return false;
		}
		if (Integer.parseInt(identityCode.substring(10, 12)) < 1
				|| Integer.parseInt(identityCode.substring(10, 12)) > 12) {// 11-12位代表出生月份，范围应该在01-12之间
			return false;
		}
		if (Integer.parseInt(identityCode.substring(12, 14)) < 1
				|| Integer.parseInt(identityCode.substring(12, 14)) > 31) {// 13-14位是出生日期，范围应该在01-31之间
			return false;
		}
		// 校验第18位
		// S = Sum(Ai * Wi), i = 0, ... , 16 ，先对前17位数字的权求和
		// Ai:表示第i位置上的身份证号码数字值
		// Wi:表示第i位置上的加权因子
		// Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
		String[] tempA = identityCode.split("|");
		int[] a = new int[18];
		for (int i = 0; i < tempA.length - 2; i++) {
			a[i] = Integer.parseInt(tempA[i + 1]);
		}
		int[] w = { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2 }; // 加权因子
		int sum = 0;
		for (int i = 0; i < 17; i++) {
			sum = sum + a[i] * w[i];
		}
		// Y = mod(S, 11)
		// 通过模得到对应的校验码
		// Y: 0 1 2 3 4 5 6 7 8 9 10
		// 校验码: 1 0 X 9 8 7 6 5 4 3 2
		String[] v = { "1", "0", "x", "9", "8", "7", "6", "5", "4", "3", "2" }; // 校验码
		
		int y = sum % 11;
		
	
		if (!v[y].equalsIgnoreCase(identityCode.substring(17))) {// 第18位校验码错误
			return false;
		}
		return true;
	}
	// 360403200501010014 17 16-18
	// 360403200701010014 15 8-16
	// 360403201601010014 6 未满8
	// 360403199501010014 27
	//


    // 身份证校验码
    private static final int[] COEFFICIENT_ARRAY = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2};

    // 身份证号的尾数规则
    private static final String[] IDENTITY_MANTISSA = {"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};

    private static final String IDENTITY_PATTERN = "^[0-9]{17}[0-9Xx]$";

    public static boolean isLegalPattern(String identity) {
        if (identity == null) {
            return false;
        }

        if (identity.length() != 18) {
            return false;
        }

        if (!identity.matches(IDENTITY_PATTERN)) {
            return false;
        }

        char[] chars = identity.toCharArray();
        long sum = IntStream.range(0, 17).map(index -> {
            char ch = chars[index];
            int digit = Character.digit(ch, 10);
            int coefficient = COEFFICIENT_ARRAY[index];
            return digit * coefficient;
        }).summaryStatistics().getSum();

        // 计算出的尾数索引
        int mantissaIndex = (int) (sum % 11);
        String mantissa = IDENTITY_MANTISSA[mantissaIndex];

        String lastChar = identity.substring(17);
        if (lastChar.equalsIgnoreCase(mantissa)) {
            return true;
        } else {
            return false;
        }
    }
}



